# CS391R1: Computer & Memory Architectures Lab 1: ALU

**Due:** Friday Sept 26, 2025 at 2:30pm ET (before Discussion Section)

#### 1. Part 1: Design Exercises (Individual Work)

Introduction

Problem 1.1: Review of Binary Addition

Problem 1.2: Designing a Combinational Circuit for Addition

Problem 1.3: Designing Sequential Logic (FSMs) to Test an Adder

#### 2. Part 2: Lab Activity (Individual Work)

Introduction

Required: Make Intermediate Github Git Commits of Your Code

Problem 2.1: Basic ALU and Testbench

Problem 2.2: NOT and XOR

Problem 2.3: Parameterizing Bit-Width

Problem 2.4: AND, OR, XNOR

Problem 2.5: Shifts

Problem 2.6: Addition and Subtraction

Problem 2.7: Comparisons

Extra Challenges: More ALU Ops

#### 3. Submission Instructions

# 1. Part 1: Design Exercises (Individual Work)

#### Introduction

We'll begin with some written problems and design exercises. You can submit your answers as a PDF (see submission instructions at the end of the document). We strongly encourage you to show your work! Showing your work and writing comments makes it much more likely that graders can award partial credit on problems.

11

8 4

All exercises in this part are to be done as **Individual** work.

# Problem 1.1: Review of Binary Addition

When building an ALU, one of the core operations it has to perform is binary addition, often on signed (positive and negative) numbers. Recall that subtraction can be performed by converting a number to negative and doing addition. Also recall that to convert between positive and negative numbers, you can "flip the bits and add one."

Calculate (by hand) the following additions using 6-bit 2's complement binary arithmetic (i.e., addition in base 2 keeping only 6 bits of your answer). Show your work, in binary notation.

(a) 
$$12+11$$
 | (b)  $13-18$  |  $3+(-18)$  | O  $10 \text{ O}10$  |

(g) Briefly explain what happened in the last addition. In what sense is your answer technically "correct"?

Problem 1.2: Designing a Combinational Circuit for Addition

One way to do binary addition is with a "ripple carry adder". The ripple carry adder strings together a chain of single-bit "full adders", sending the carry-out signal from the lower-order bits to the higher-order bits. xor



Fig. 1. Full adder and a combinational logic circuit implementing the full adder.



Fig. 2. Ripple carry adders: (a) two-bit adder, and (b) four-bit adder.

(a) Please fill in the missing truth table values below for the two-bit ripple carry adder (Fig. 2a). We will assume that C0 is permanently set to "0". The table is divided into two parts.

| A1 | A0 | B1 | В0 | C2 | S1 | S0 |   | A1 | A0 | B1 | В0 | C2 | S1 | S0 |     |
|----|----|----|----|----|----|----|---|----|----|----|----|----|----|----|-----|
| 0  | 0  | 0  | 0  | 0  | 0  | 0  |   | 1  | 0  | 0  | 0  | 0  | 1  | 0  |     |
| 0  | 0  | 0  | 1  | 0  | 0  | 1  |   | 1  | 0  | 0  | 1  | 0  | 1  | 1  |     |
| 0  | 0  | 1  | 0  | 0  | 1  | 0  | ← | 1  | 0  | 1  | 0  | 1  | 0  | 0  |     |
| 0  | 0  | 1  | 1  | 0  | 1  | 1  | ← | 1  | 0  | 1  | 1  | 0  | O  | 0  | ←   |
| 0  | 1  | 0  | 0  | 0  | 0  | 1  |   | 1  | 1  | 0  | 0  | 0  | 1  | 1  |     |
| 0  | 1  | 0  | 1  | 0  | 1  | 1  | ← | 1  | 1  | 0  | 1  | 1  | 0  | 0  | ←   |
| 0  | 1  | 1  | 0  | 0  | 1  | 1  |   | 1  | 1  | 1  | 0  | 1  | 0  | 1  | ← - |
| 0  | 1  | 1  | 1  | 0  | 0  | 0  | ← | 1  | 1  | 1  | 1  | 1  | 1  | 0  | ←   |

Say that we assemble an 8-bit ripple carry adder out of two 4-bit ripple carry adders as pictured below. Assume that the 4-bit adders are assembled from the single-bit full adder circuit pictured above in Fig. 1.



Fig. 3. An (a) 8-bit ripple carry adder, made from two cascaded (b) 4-bit adders (see Fig. 2b).

(b) How many total 1-input or 2-input logic gates (e.g., 1-input NOT gates, 2-input XOR gates, 2-input MUX gates) are required to build the 8-bit ripple carry adder design?

(c) What is the tpd of the 8-bit ripple carry adder? Show your work. What was your approach to find this answer?



(d) What is the tcd of the 8-bit ripple carry adder? Show your work. What was your approach to

Approach: took the min time it takes for input to flow through circuit find this answer?



Here's a second potential design for the 8-bit adder called a "carry-select adder" (see Fig. 4). It is assembled from 4-bit ripple carry adders (see Fig. 2b) and 2-input muxes.



Fig. 4. An 8-bit carry-select adder, assembled from 4-bit ripple-carry adders (see Fig. 2b).

(e) In words, briefly describe what you think this circuit does, and how it might work.

The diagram above performs 8 bit addition using a 4 bit full adder to calculate the first lower 4 bits of the result and then two pre-computed other 4 bit full adders to obtain the last upper 4 bits of the result. Based off of whether the output of the first 4 bit full adder is 1 or 0, the carry-select adder will choose the appropriate area result given the input.

(f) How many total 1-input or 2-input logic gates (e.g., 1-input NOT gates, 2-input XOR gates, 2-input MUX gates) are required to build the 8-bit carry-select adder design?

(g) What is the tpd of the 8-bit carry-select adder? Show your work.

$$95 + 95 + 95 + 50 \rightarrow 235_{ns} + 10_{ns}(*9) \rightarrow 275_{ns}$$
1 2 3 9 MUX

(h) What is the tcd of the 8-bit carry-select adder? Show your work.

(i) What do you think might be potential advantages of the carry-select adder design versus the ripple-carry adder design?

The carry-select adder design allows us to plot out how in a different way how the ripple-carry adder functions. Consequently, it reduces the time for our propagation delay, allowing for more optimized performance.

(j) What do you think might be potential disadvantages of the carry-select adder design versus the ripple-carry adder design?

A potential disadvantage of the carry-select adder design could be that it takes more power to run the entire circuit even though it stills runs at a faster rate compared to the ripple-carry adder design which has a more simple architecture.

## Problem 1.3: Designing Sequential Logic (FSMs) to Test an Adder

Let's say that we want to allow users to perform a sequence of additions on a combinational 2-bit ripple carry adder.



Fig. 5. A 2-bit ripple carry adder (see Fig. 2a).

We would like to use finite state machines to perform the following six additions in sequence:

0+0,

1+0,

1+0.

1+1,

2+1,

3+2

We design the following device with sequential logic. It has an input button for the user to press to move on to the next addition in the sequence (see above). This input button connects to the "input generator" finite state machine (FSM), FSM\_inputgen. The outputs of FSM\_inputgen connect to the input signals of the 2-bit ripple carry adder. The outputs of the 2-bit ripple carry adder then are given as inputs to a second "output checker" FSM, FSM\_outputcheck. The output of FSM\_outputcheck is a binary "success" signal that is "1" when the user successfully completes all six desired additions and their outputs are correct, and is "0" otherwise.



Fig. 6. Diagram of FSM-based testing system for 2-bit adder.

(a) Draw a valid Moore machine FSM diagram for the FSM inputgen. We have given the BP:0 starting state below: BP:0 3 4 1 (Start here on reset.) a[1:0]:2'601 Button Start BP:1 06.5:[0:1]0 a[1:0]:2601 BP:1 Press = 1 a[1:0] = 2'b00Z 1[1:0]:2,901 a[1:0]:2'100 6[1:0]:2600 b[1:0] = 2'b00 Cin = 1'10 Cin = 1'b0 Cin = 1'60 Cin=1.10 Button Press = 0 BP: 0 BP:1 L ζ **BP**: 0 a[1:0]: 2'611 a[1:0]:2610 BP:1 6[1:0]: 2'610 6[1:0]: 2'601 Cin = 1'60 Cin = 1'60 gp:0

For the diagram above, it depicts the transition between each state. For the 2nd, 4th, 5th, and 6th state, they would all proceed to the next state after a button press which changes the condition. No button press for these states will not change these addition expressions at all. However, for the 3rd state, it is a duplicate state and when the button press is 0 (null) the 3rd state reverts back to state 2.

(b) Draw a valid Moore machine FSM diagram for the FSM\_outputcheck. We have given the starting state below:



This would depend on the carry out value and the final output of the two inputs. There would be overflow at the very end with our two operations of 2+1 and 3+2. At the end here our else would go back to the 3rd success beacuse they both return 1.

(c) Now, say that someone gives you a broken 2-bit adder, where the S1 output signal is stuck always outputting a "0". What happens when a user tries to use the input button and step through the six addition operations above? Describe your answer in terms of the state transitions and signals in your FSM\_inputgen and FSM\_outputcheck diagrams above.

The user would keep getting a success value of 0 because in our finite state diagram we see that when there is no button press our addition operation is never changed. T

# 2. Part 2: Lab Activity (Individual Work)

#### Introduction

In this lab activity, we will build an ALU (Arithmetic and Logic Unit). The ALU is a crucial part of any processor architecture. It is a circuit that is responsible for performing arithmetic and logic operations. We will start with a very simple design and gradually build more functionality into it. By the end of this lab, we should have an ALU as a standalone module which supports numerous logical (NOT, AND, OR, XOR, XNOR, UNSIGNED SHIFT) and arithmetic (ADD, SUBTRACT, LESS, EQUALS, GREATER, SIGNED SHIFT) operations on multi-bit inputs. The ALU that you build in this lab will be an important building block for future assignments, so make sure you fully understand how it works and that you are able to debug and modify it.

All exercises in this part are to be done as **Individual** work.

## Required: Make Intermediate Github Git Commits of Your Code

In modern digital design, we are often working with other engineers in large teams, and making careful modifications to a complex larger system. This makes it critical to document the step-by-step changes that we make to a Verilog code base through a series of git commits with comments as we work. Frequent and well-commented git commits can also save progress and prevent having to start over in the event that local data becomes corrupted.

To encourage best coding practices, in this class we will submit code for our lab activities by submitting private Github repositories with the required files in them. We will be looking for a record of intermediate git commits of work-in-progress (with comments), and this will be part of the grade for the assignment. So, remember: Commit early, commit often!

#### Problem 2.1: Basic ALU and Testbench

First, we start with something very simple. Let's create a SystemVerilog module (you can call it alu.sv) that takes in one single-bit input wire (call it op1 which stands for operand 1) and produces a NOT of it (call the output wire res, which stands for result).

Create a testbench file (call it alu\_tb.sv) in the Simulation Sources directory. Write tests to make sure your module works as desired.

#### Problem 2.2: NOT and XOR

Now, let's expand the functionality of our module. Introduce two new single-bit input wires for a second operand and control (call them op2 and control).

Update your ALU code to behave as follows:

- If control wire is LOW, res should be a NOT of op1. Same behavior as before.
- If control wire is HIGH, res should be an XOR of op1 and op2.

Update your testbench to verify that your module works correctly.

## Problem 2.3: Parameterizing Bit-Width

So far we have worked with single-bit inputs to our ALU. Let's introduce a parameter (call it OP WIDTH) which can be set when the module is instantiated. This parameter will determine the bit length of the operands and the result. Make sure that both of the operations still work correctly (NOT and XOR should be bit-wise operations).

Update your testbench.

## Problem 2.4: AND, OR, XNOR

Let's update our module to include the remaining logical operations (AND, OR, XNOR).

#### Hints:

- When you do this, the first issue that you'll encounter is that our control wire is a single bit, so it can only represent two operations. Think about how you can overcome this.
- Once you have figured out the control wire issue, you might notice that in some cases our ALU has an undefined behavior. Introduce a new single-bit output wire called error. This wire should be raised HIGH for inputs that either result in a computational error (e.g. division by zero) or do not have a defined behavior. Otherwise, it should be LOW. From now on, when you update your ALU, always make sure that you properly set this error output.

Update your testbench.

#### Problem 2.5: Shifts

Update the ALU to include logical (UNSIGNED) and arithmetic (SIGNED) bit-wise shifts. op1 will be the input on which we want to perform the shift. op2 will be the input that determines by how many bits we shift.

Update your testbench.

#### Problem 2.6: Addition and Subtraction

Update the ALU to include simple arithmetic operations: addition and subtraction. To have some common convention, in the case of subtraction set res = op1 - op2.

Update your testbench.

## Problem 2.7: Comparisons

Update the ALU to include comparison operations: less than, equals to, greater than. The first bit of the result output determines whether the comparison is true (1'b1) or false (1'b0). Other bits of the result should be set to 0.

To have some common convention, make sure the op1 is always on the left side of the operation. For example, if we do less than, res = op1 < op2.

Update your test-bench.

# Extra Challenges: More ALU Ops

Congratulations, you have completed the required part of the lab! However, we don't want to let you off so easily, so we prepared some additional challenges for you. All of these challenges are completely optional, however, we strongly encourage you to try to do some of them as they can be very useful for future assignments.

Here is the list of challenges in the order of increasing difficulty:

- Introduce the remaining logical operations: NAND, NOR, XNOR.
- Introduce not bit-wise logical operations: AND, OR, NOT. They should behave the same as &&, ||, ! do in the C programming language.
- Complete the comparison functionality of your ALU. Introduce greater than or equal to, less than or equal to, not equal to operations.

- Complete the arithmetic functionality of your ALU. Introduce multiplication, integer division, modulus, and exponent operations.
- Create a new module that takes in a single 4-byte input and produces a boolean output which determines if at least one of the bytes in the input is zero. Hint: use your ALU module inside the new module.

Don't forget to update your testbench.

## 3. Submission Instructions

We will use Gradescope for all submissions. See detailed instructions below.

**Design Exercises:** Submit your written solutions to the Lab 1 Design Exercises as a PDF file. We strongly encourage you to show your work! Showing your work and writing comments makes it much more likely that graders can award partial credit on problems.

**Source Code:** You should submit your complete source code which includes any of the module sources and testbenches.

Follow the same instructions as Lab 0:

- 1. Create a **private** GitHub repository.
- 2. Push your code to that repository. Remember to make the required intermediate git commits of work-in-progress as you go!
- 3. These are all **Individual** exercises in Lab 1, so please submit your own work. You can submit directly from your GitHub repository. If you are having difficulties submitting your code to Gradescope from GitHub, please see the TF at Discussion or office hours, and let us know as soon as possible before the deadline, to have time to debug.

**Lab Report:** In addition to that, you should include a concise report (as a PDF file) of how you approached the problems, what challenges you encountered, and if you have any questions and/or recommendations. There is no need to write an in-depth report, we just want to see that you have a decent understanding of what you implemented.

#### **Summary of Deliverables:**

- Design Exercises PDF file
- Source Code
  - alu.sv
  - o alu tb.sv
  - o ... and any other supporting source code or files, as needed
  - Remember to make the required intermediate git commits as you go!
- Lab Report PDF file